"""Test for the LCN light platform."""

from unittest.mock import patch

from freezegun.api import FrozenDateTimeFactory
from pypck.inputs import ModStatusOutput, ModStatusRelays
from pypck.lcn_addr import LcnAddr
from pypck.lcn_defs import RelayStateModifier
import pytest
from syrupy.assertion import SnapshotAssertion

from homeassistant.components.lcn.helpers import get_device_connection
from homeassistant.components.lcn.light import SCAN_INTERVAL
from homeassistant.components.light import (
    ATTR_BRIGHTNESS,
    ATTR_TRANSITION,
    DOMAIN as DOMAIN_LIGHT,
)
from homeassistant.const import (
    ATTR_ENTITY_ID,
    SERVICE_TURN_OFF,
    SERVICE_TURN_ON,
    STATE_OFF,
    STATE_ON,
    STATE_UNAVAILABLE,
    Platform,
)
from homeassistant.core import HomeAssistant
from homeassistant.helpers import entity_registry as er

from .conftest import MockConfigEntry, MockDeviceConnection, init_integration

from tests.common import async_fire_time_changed, snapshot_platform

LIGHT_OUTPUT1 = "light.testmodule_light_output1"
LIGHT_OUTPUT2 = "light.testmodule_light_output2"
LIGHT_RELAY1 = "light.testmodule_light_relay1"


async def test_setup_lcn_light(
    hass: HomeAssistant,
    entity_registry: er.EntityRegistry,
    entry: MockConfigEntry,
    snapshot: SnapshotAssertion,
) -> None:
    """Test the setup of light."""
    with patch("homeassistant.components.lcn.PLATFORMS", [Platform.LIGHT]):
        await init_integration(hass, entry)

    await snapshot_platform(hass, entity_registry, snapshot, entry.entry_id)


async def test_output_turn_on(hass: HomeAssistant, entry: MockConfigEntry) -> None:
    """Test the output light turns on."""
    await init_integration(hass, entry)

    with patch.object(MockDeviceConnection, "toggle_output") as toggle_output:
        # command failed
        toggle_output.return_value = False

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_OUTPUT1},
            blocking=True,
        )

        toggle_output.assert_awaited_with(0, 9, to_memory=True)

        state = hass.states.get(LIGHT_OUTPUT1)
        assert state is not None
        assert state.state != STATE_ON

        # command success
        toggle_output.reset_mock(return_value=True)
        toggle_output.return_value = True

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_OUTPUT1},
            blocking=True,
        )

        toggle_output.assert_awaited_with(0, 9, to_memory=True)

        state = hass.states.get(LIGHT_OUTPUT1)
        assert state is not None
        assert state.state == STATE_ON


async def test_output_turn_on_with_attributes(
    hass: HomeAssistant, entry: MockConfigEntry
) -> None:
    """Test the output light turns on."""
    await init_integration(hass, entry)

    with patch.object(MockDeviceConnection, "dim_output") as dim_output:
        dim_output.return_value = True

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {
                ATTR_ENTITY_ID: LIGHT_OUTPUT1,
                ATTR_BRIGHTNESS: 50,
                ATTR_TRANSITION: 2,
            },
            blocking=True,
        )

        dim_output.assert_awaited_with(0, 19, 6)

        state = hass.states.get(LIGHT_OUTPUT1)
        assert state is not None
        assert state.state == STATE_ON


async def test_output_turn_off(hass: HomeAssistant, entry: MockConfigEntry) -> None:
    """Test the output light turns off."""
    await init_integration(hass, entry)

    with patch.object(MockDeviceConnection, "toggle_output") as toggle_output:
        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_OUTPUT1},
            blocking=True,
        )

        # command failed
        toggle_output.return_value = False

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_OFF,
            {ATTR_ENTITY_ID: LIGHT_OUTPUT1},
            blocking=True,
        )

        toggle_output.assert_awaited_with(0, 9, to_memory=True)

        state = hass.states.get(LIGHT_OUTPUT1)
        assert state is not None
        assert state.state != STATE_OFF

        # command success
        toggle_output.reset_mock(return_value=True)
        toggle_output.return_value = True

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_OFF,
            {ATTR_ENTITY_ID: LIGHT_OUTPUT1},
            blocking=True,
        )

        toggle_output.assert_awaited_with(0, 9, to_memory=True)

        state = hass.states.get(LIGHT_OUTPUT1)
        assert state is not None
        assert state.state == STATE_OFF


async def test_relay_turn_on(hass: HomeAssistant, entry: MockConfigEntry) -> None:
    """Test the relay light turns on."""
    await init_integration(hass, entry)

    with patch.object(MockDeviceConnection, "control_relays") as control_relays:
        states = [RelayStateModifier.NOCHANGE] * 8
        states[0] = RelayStateModifier.ON

        # command failed
        control_relays.return_value = False

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_RELAY1},
            blocking=True,
        )

        control_relays.assert_awaited_with(states)

        state = hass.states.get(LIGHT_RELAY1)
        assert state is not None
        assert state.state != STATE_ON

        # command success
        control_relays.reset_mock(return_value=True)
        control_relays.return_value = True

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_RELAY1},
            blocking=True,
        )

        control_relays.assert_awaited_with(states)

        state = hass.states.get(LIGHT_RELAY1)
        assert state is not None
        assert state.state == STATE_ON


async def test_relay_turn_off(hass: HomeAssistant, entry: MockConfigEntry) -> None:
    """Test the relay light turns off."""
    await init_integration(hass, entry)

    with patch.object(MockDeviceConnection, "control_relays") as control_relays:
        states = [RelayStateModifier.NOCHANGE] * 8
        states[0] = RelayStateModifier.OFF

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_ON,
            {ATTR_ENTITY_ID: LIGHT_RELAY1},
            blocking=True,
        )

        # command failed
        control_relays.return_value = False

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_OFF,
            {ATTR_ENTITY_ID: LIGHT_RELAY1},
            blocking=True,
        )

        control_relays.assert_awaited_with(states)

        state = hass.states.get(LIGHT_RELAY1)
        assert state is not None
        assert state.state != STATE_OFF

        # command success
        control_relays.reset_mock(return_value=True)
        control_relays.return_value = True

        await hass.services.async_call(
            DOMAIN_LIGHT,
            SERVICE_TURN_OFF,
            {ATTR_ENTITY_ID: LIGHT_RELAY1},
            blocking=True,
        )

        control_relays.assert_awaited_with(states)

        state = hass.states.get(LIGHT_RELAY1)
        assert state is not None
        assert state.state == STATE_OFF


async def test_pushed_output_status_change(
    hass: HomeAssistant, entry: MockConfigEntry
) -> None:
    """Test the output light changes its state on status received."""
    await init_integration(hass, entry)

    device_connection = get_device_connection(hass, (0, 7, False), entry)
    address = LcnAddr(0, 7, False)

    # push status "on"
    inp = ModStatusOutput(address, 0, 50)
    await device_connection.async_process_input(inp)
    await hass.async_block_till_done()

    state = hass.states.get(LIGHT_OUTPUT1)
    assert state is not None
    assert state.state == STATE_ON
    assert state.attributes[ATTR_BRIGHTNESS] == 128

    # push status "off"
    inp = ModStatusOutput(address, 0, 0)
    await device_connection.async_process_input(inp)
    await hass.async_block_till_done()

    state = hass.states.get(LIGHT_OUTPUT1)
    assert state is not None
    assert state.state == STATE_OFF


async def test_pushed_relay_status_change(
    hass: HomeAssistant, entry: MockConfigEntry
) -> None:
    """Test the relay light changes its state on status received."""
    await init_integration(hass, entry)

    device_connection = get_device_connection(hass, (0, 7, False), entry)
    address = LcnAddr(0, 7, False)
    states = [False] * 8

    # push status "on"
    states[0] = True
    inp = ModStatusRelays(address, states)
    await device_connection.async_process_input(inp)
    await hass.async_block_till_done()

    state = hass.states.get(LIGHT_RELAY1)
    assert state is not None
    assert state.state == STATE_ON

    # push status "off"
    states[0] = False
    inp = ModStatusRelays(address, states)
    await device_connection.async_process_input(inp)
    await hass.async_block_till_done()

    state = hass.states.get(LIGHT_RELAY1)
    assert state is not None
    assert state.state == STATE_OFF


@pytest.mark.parametrize(
    ("entity_id", "request_method", "return_value"),
    [
        (
            LIGHT_OUTPUT1,
            "request_status_output",
            ModStatusOutput(LcnAddr(0, 7, False), 0, 0),
        ),
        (
            LIGHT_RELAY1,
            "request_status_relays",
            ModStatusRelays(LcnAddr(0, 7, False), [False] * 8),
        ),
    ],
)
async def test_availability(
    hass: HomeAssistant,
    freezer: FrozenDateTimeFactory,
    entry: MockConfigEntry,
    entity_id: str,
    request_method: str,
    return_value: ModStatusOutput | ModStatusRelays,
) -> None:
    """Test the availability of light entity."""
    await init_integration(hass, entry)

    state = hass.states.get(entity_id)
    assert state is not None
    assert state.state != STATE_UNAVAILABLE

    # no response from device -> unavailable
    with patch.object(MockDeviceConnection, request_method, return_value=None):
        freezer.tick(SCAN_INTERVAL)
        async_fire_time_changed(hass)
        await hass.async_block_till_done(wait_background_tasks=True)

    state = hass.states.get(entity_id)
    assert state is not None
    assert state.state == STATE_UNAVAILABLE

    # response from device -> available
    with patch.object(
        MockDeviceConnection,
        request_method,
        return_value=return_value,
    ):
        freezer.tick(SCAN_INTERVAL)
        async_fire_time_changed(hass)
        await hass.async_block_till_done(wait_background_tasks=True)

    state = hass.states.get(entity_id)
    assert state is not None
    assert state.state != STATE_UNAVAILABLE


async def test_unload_config_entry(hass: HomeAssistant, entry: MockConfigEntry) -> None:
    """Test the light is removed when the config entry is unloaded."""
    await init_integration(hass, entry)

    await hass.config_entries.async_unload(entry.entry_id)
    assert hass.states.get(LIGHT_OUTPUT1).state == STATE_UNAVAILABLE
